package main

import (
	"fmt"
	"time"
)

// 在这个例子中，我们将看到如何使用 Go 协程和通道 实现一个工作池 。

// 这是我们将要在多个并发实例中支持的任务了。这些执行者 将从 jobs 通道接收任务，并且通过 results 发送对应 的结果。我们将让每个任务间隔 1s 来模仿一个耗时的任务。
func worker(id int, jobs <-chan int, result chan<- int) {
	for j := range jobs {
		fmt.Println("worker", id, "processing job", j)
		time.Sleep(time.Second)
		result <- j * 2
	}
	close(result)
}

func main() {
	// 为了使用 worker 工作池并且收集他们的结果，我们需要 2 个通道。
	jobs := make(chan int, 100)
	result := make(chan int, 100)

	// 这里启动了 3 个 worker，初始是阻塞的，因为 还没有传递任务。
	for w := 1; w <= 3; w++ {
		go worker(w, jobs, result)
	}
	// 这里我们发送 9 个 jobs，然后 close 这些通道 来表示这些就是所有的任务了。
	for j := 1; j <= 9; j++ {
		jobs <- j
	}
	close(jobs)

	// 最后，我们收集所有这些任务的返回值。
	for a := 1; a <= 9; a++ {
		<-result
	}
	// for a := range result {
	// 	println(a)
	// }

	// 执行这个程序，显示 9 个任务被多个 worker 执行。整个程序 处理所有的任务仅执行了 3s 而不是 9s，是因为 3 个 worker 是并行的。
}
